home *** CD-ROM | disk | FTP | other *** search
/ CU Amiga Super CD-ROM 15 / CU Amiga Magazine's Super CD-ROM 15 (1997)(EMAP Images)(GB)[!][issue 1997-10].iso / CUCD / Graphics / Ghostscript / source / zcrd.c < prev    next >
C/C++ Source or Header  |  1997-04-24  |  10KB  |  312 lines

  1. /* Copyright (C) 1995, 1996, 1997 Aladdin Enterprises.  All rights reserved.
  2.   
  3.   This file is part of Aladdin Ghostscript.
  4.   
  5.   Aladdin Ghostscript is distributed with NO WARRANTY OF ANY KIND.  No author
  6.   or distributor accepts any responsibility for the consequences of using it,
  7.   or for whether it serves any particular purpose or works at all, unless he
  8.   or she says so in writing.  Refer to the Aladdin Ghostscript Free Public
  9.   License (the "License") for full details.
  10.   
  11.   Every copy of Aladdin Ghostscript must include a copy of the License,
  12.   normally in a plain ASCII text file named PUBLIC.  The License grants you
  13.   the right to copy, modify and redistribute Aladdin Ghostscript, but only
  14.   under certain conditions described in the License.  Among other things, the
  15.   License requires that the copyright notice and this notice be preserved on
  16.   all copies.
  17. */
  18.  
  19. /* zcrd.c */
  20. /* CIE color rendering operators */
  21. #include "math_.h"
  22. #include "ghost.h"
  23. #include "errors.h"
  24. #include "oper.h"
  25. #include "gsstruct.h"
  26. #include "gscspace.h"
  27. #include "gscolor2.h"
  28. #include "gscie.h"
  29. #include "estack.h"
  30. #include "ialloc.h"
  31. #include "idict.h"
  32. #include "idparam.h"
  33. #include "igstate.h"
  34. #include "icie.h"
  35. #include "isave.h"
  36. #include "ivmspace.h"
  37. #include "store.h"        /* for make_null */
  38.  
  39. /* Forward references */
  40. private int zcrd_params(P4(os_ptr, gs_cie_render *, ref_cie_render_procs *, gs_memory_t *));
  41. private int cache_colorrendering(P3(gs_cie_render *,
  42.   const ref_cie_render_procs *, gs_state *));
  43.  
  44. /* Allocator structure type for CIE rendering structure */
  45. private_st_cie_render();
  46.  
  47. /* - currentcolorrendering <dict> */
  48. private int
  49. zcurrentcolorrendering(register os_ptr op)
  50. {    push(1);
  51.     *op = istate->colorrendering.dict;
  52.     return 0;
  53. }
  54.  
  55. /* <dict> setcolorrendering - */
  56. private int
  57. zsetcolorrendering(register os_ptr op)
  58. {    gs_memory_t *mem = gs_state_memory(igs);
  59.     int code;
  60.     es_ptr ep = esp;
  61.     gs_cie_render *pcie;
  62.     ref_cie_render_procs procs_old;
  63.  
  64.     check_read_type(*op, t_dictionary);
  65.     check_dict_read(*op);
  66.     rc_alloc_struct_0(pcie, gs_cie_render, &st_cie_render, mem,
  67.               return_error(e_VMerror),
  68.               "setcolorrendering");
  69.     /* gs_setcolorrendering may refer to istate->colorrendering.procs. */
  70.     procs_old = istate->colorrendering.procs;
  71.     code = zcrd_params(op, pcie, &istate->colorrendering.procs, mem);
  72.     if ( code < 0 ||
  73.          (code = gs_setcolorrendering(igs, pcie)) < 0 ||
  74.          (code = cache_colorrendering(pcie, &istate->colorrendering.procs, igs)) < 0
  75.        )
  76.     {    rc_free_struct(pcie, "setcolorrendering");
  77.         istate->colorrendering.procs = procs_old;
  78.         esp = ep;
  79.         return code;
  80.     }
  81.     istate->colorrendering.dict = *op;
  82.     pop(1);
  83.     return (esp == ep ? 0 : o_push_estack);
  84. }
  85. /* Get the CRD parameters from the PostScript dictionary. */
  86. private int
  87. zcrd_params(os_ptr op, gs_cie_render *pcie,
  88.   ref_cie_render_procs *pcprocs, gs_memory_t *mem)
  89. {    int code;
  90.     int ignore;
  91.     ref *pRT;
  92.  
  93.     if ( (code = dict_int_param(op, "ColorRenderingType", 1, 1, 0, &ignore)) < 0 ||
  94.          (code = dict_matrix3_param(op, "MatrixLMN", &pcie->MatrixLMN)) != matrix3_ok ||
  95.          (code = dict_proc3_param(op, "EncodeLMN", &pcprocs->EncodeLMN)) < 0 ||
  96.          (code = dict_range3_param(op, "RangeLMN", &pcie->RangeLMN)) < 0 ||
  97.          (code = dict_matrix3_param(op, "MatrixABC", &pcie->MatrixABC)) != matrix3_ok ||
  98.          (code = dict_proc3_param(op, "EncodeABC", &pcprocs->EncodeABC)) < 0 ||
  99.          (code = dict_range3_param(op, "RangeABC", &pcie->RangeABC)) < 0 ||
  100.          (code = cie_points_param(op, &pcie->points)) < 0 ||
  101.          (code = dict_matrix3_param(op, "MatrixPQR", &pcie->MatrixPQR)) != matrix3_ok ||
  102.          (code = dict_range3_param(op, "RangePQR", &pcie->RangePQR)) < 0 ||
  103.          (code = dict_proc3_param(op, "TransformPQR", &pcprocs->TransformPQR)) < 0
  104.        )
  105.       return (code < 0 ? code : gs_note_error(e_rangecheck));
  106. #define rRT pcie->RenderTable.lookup
  107.     if ( dict_find_string(op, "RenderTable", &pRT) > 0 )
  108.     {    const ref *prte;
  109.         int i;
  110.  
  111.         check_read_type(*pRT, t_array);
  112.         if ( r_size(pRT) < 5 )
  113.           return_error(e_rangecheck);
  114.         prte = pRT->value.const_refs;
  115.         check_type_only(prte[4], t_integer);
  116.         if ( !(prte[4].value.intval == 3 || prte[4].value.intval == 4) )
  117.           return_error(e_rangecheck);
  118.         rRT.n = 3;
  119.         rRT.m = prte[4].value.intval;
  120.         if ( r_size(pRT) != rRT.m + 5 )
  121.           return_error(e_rangecheck);
  122.         prte += 5;
  123.         for ( i = 0; i < rRT.m; i++ )
  124.           check_proc_only(prte[i]);
  125.         code = cie_table_param(pRT, &rRT, mem);
  126.         if ( code < 0 )
  127.           return code;
  128.         make_const_array(&pcprocs->RenderTableT,
  129.                  a_readonly | r_space(pRT),
  130.                  rRT.m, prte);
  131.     }
  132.     else
  133.     {    rRT.table = 0;
  134.         make_null(&pcprocs->RenderTableT);
  135.     }
  136. #undef rRT
  137.     pcie->EncodeLMN = Encode_default;
  138.     pcie->EncodeABC = Encode_default;
  139.     pcie->TransformPQR = TransformPQR_default;
  140.     pcie->RenderTable.T = RenderTableT_default;
  141.     return 0;
  142. }
  143.  
  144. /* Cache the results of the color rendering procedures. */
  145. private int cie_cache_render_finish(P1(os_ptr));
  146. private int
  147. cache_colorrendering(gs_cie_render *pcie,
  148.   const ref_cie_render_procs *pcrprocs, gs_state *pgs)
  149. {    es_ptr ep = esp;
  150.     int code = gs_cie_render_init(pcie);    /* sets Domain values */
  151.     int i;
  152.  
  153.     /* We must run gs_cie_render_complete when we're done. */
  154.     if ( code < 0 ||
  155.          (gs_cie_cs_common(pgs) != 0 &&
  156.           (code = cie_cache_joint(pcrprocs, pgs)) < 0) ||    /* do this last */
  157.          (code = cie_cache_push_finish(cie_cache_render_finish, pgs, pcie)) < 0 ||
  158.          (code = cie_prepare_cache3(&pcie->DomainLMN, pcrprocs->EncodeLMN.value.const_refs, &pcie->caches.EncodeLMN[0], pcie, pgs, "Encode.LMN")) < 0 ||
  159.          (code = cie_prepare_cache3(&pcie->DomainABC, pcrprocs->EncodeABC.value.const_refs, &pcie->caches.EncodeABC[0], pcie, pgs, "Encode.ABC")) < 0
  160.        )
  161.       {    esp = ep;
  162.         return code;
  163.       }
  164.     if ( pcie->RenderTable.lookup.table != 0 )
  165.       { bool is_identity = true;
  166.         for ( i = 0; i < pcie->RenderTable.lookup.m; i++ )
  167.           if ( r_size(pcrprocs->RenderTableT.value.const_refs + i) != 0 )
  168.         { is_identity = false;
  169.           break;
  170.         }
  171.         pcie->caches.RenderTableT_is_identity = is_identity;
  172.         if ( !is_identity )
  173.           for ( i = 0; i < pcie->RenderTable.lookup.m; i++ )
  174.         if ( (code =
  175.               cie_prepare_cache(Range4_default.ranges,
  176.                     pcrprocs->RenderTableT.value.const_refs + i,
  177.                     &pcie->caches.RenderTableT[i].floats,
  178.                     pcie, pgs, "RenderTable.T")) < 0
  179.            )
  180.           { esp = ep;
  181.             return code;
  182.           }
  183.       }
  184.     return o_push_estack;
  185. }
  186.  
  187. /* Finish up after loading the rendering caches. */
  188. private int
  189. cie_cache_render_finish(os_ptr op)
  190. {    gs_cie_render *pcie = r_ptr(op, gs_cie_render);
  191.     int code;
  192.     if ( pcie->RenderTable.lookup.table != 0 && !pcie->caches.RenderTableT_is_identity )
  193.     {    /* Convert the RenderTableT cache from floats to fracs. */
  194.         int j;
  195.         for ( j = 0; j < pcie->RenderTable.lookup.m; j++ )
  196.           gs_cie_cache_to_fracs(&pcie->caches.RenderTableT[j]);
  197.     }
  198.     code = gs_cie_render_complete(pcie);
  199.     if ( code < 0 )
  200.       return code;
  201.     pop(1);
  202.     return 0;
  203. }
  204.  
  205. /* ------ Internal routines ------ */
  206.  
  207. /* Load the joint caches. */
  208. private int
  209.   cie_exec_tpqr(P1(os_ptr)),
  210.   cie_post_exec_tpqr(P1(os_ptr)),
  211.   cie_tpqr_finish(P1(os_ptr));
  212. int
  213. cie_cache_joint(const ref_cie_render_procs *pcrprocs, gs_state *pgs)
  214. {    const gs_cie_render *pcier = gs_currentcolorrendering(pgs);
  215.     /* The former installation procedures have allocated */
  216.     /* the joint caches and filled in points_sd. */
  217.     gx_cie_joint_caches *pjc = gx_currentciecaches(pgs);
  218.     ref pqr_procs;
  219. #define pqr_refs pqr_procs.value.refs
  220.     uint space;
  221.     int code;
  222.     int i;
  223.  
  224.     if ( pcier == 0 )    /* cache is not set up yet */
  225.       return 0;
  226.     if ( pjc == 0 )
  227.       return_error(e_VMerror);
  228.     code = ialloc_ref_array(&pqr_procs, a_readonly, 3*(1+4+4*6),
  229.                 "cie_cache_common");
  230.     if ( code < 0 ) return code;
  231.     /* When we're done, deallocate the procs and complete the caches. */
  232.     check_estack(3);
  233.     cie_cache_push_finish(cie_tpqr_finish, pgs, pgs);
  234.     *++esp = pqr_procs;
  235.     space = r_space(&pqr_procs);
  236.     for ( i = 0; i < 3; i++ )
  237.     {    ref *p = pqr_refs + 3 + (4+4*6) * i;
  238.         const float *ppt = (float *)&pjc->points_sd;
  239.         int j;
  240.  
  241.         make_array(pqr_refs + i, a_readonly | a_executable | space,
  242.                4, p);
  243.         make_array(p, a_readonly | space, 4*6, p + 4);
  244.         p[1] = pcrprocs->TransformPQR.value.refs[i];
  245.         make_oper(p + 2, 0, cie_exec_tpqr);
  246.         make_oper(p + 3, 0, cie_post_exec_tpqr);
  247.         for ( j = 0, p += 4; j < 4*6; j++, p++, ppt++ )
  248.           make_real(p, *ppt);
  249.     }
  250.     return cie_prepare_cache3(&pcier->RangePQR,
  251.                   pqr_procs.value.const_refs,
  252.                   &pjc->TransformPQR[0],
  253.                   pjc, pgs, "Transform.PQR");
  254. }
  255.  
  256. /* Private operator to shuffle arguments for the TransformPQR procedure: */
  257. /* v [ws wd bs bd] proc -> -mark- ws wd bs bd v proc + exec */
  258. private int
  259. cie_exec_tpqr(register os_ptr op)
  260. {    const ref *ppt = op[-1].value.const_refs;
  261.     uint space = r_space(op - 1);
  262.     int i;
  263.  
  264.     check_op(4);
  265.     push(4);
  266.     *op = op[-4];        /* proc */
  267.     op[-1] = op[-6];    /* v */
  268.     for ( i = 0; i < 4; i++ )
  269.       make_const_array(op - 5 + i, a_readonly | space,
  270.                6, ppt + i * 6);
  271.     make_mark(op - 6);
  272.     return zexec(op);
  273. }
  274.  
  275. /* Remove extraneous values from the stack after executing */
  276. /* the TransformPQR procedure.  -mark- ... v -> v */
  277. private int
  278. cie_post_exec_tpqr(register os_ptr op)
  279. {    uint count = ref_stack_counttomark(&o_stack);
  280.     ref vref;
  281.  
  282.     if ( count < 2 )
  283.       return_error(e_unmatchedmark);
  284.     vref = *op;
  285.     ref_stack_pop(&o_stack, count - 1);
  286.     *osp = vref;
  287.     return 0;
  288. }
  289.  
  290. /* Free the procs array and complete the joint caches. */
  291. private int
  292. cie_tpqr_finish(register os_ptr op)
  293. {    gs_state *pgs = r_ptr(op, gs_state);
  294.     ifree_ref_array(op - 1, "cie_tpqr_finish");
  295.     gs_cie_cs_complete(pgs, false);
  296.     pop(2);
  297.     return 0;
  298. }
  299.  
  300. /* ------ Initialization procedure ------ */
  301.  
  302. BEGIN_OP_DEFS(zcrd_l2_op_defs) {
  303.         op_def_begin_level2(),
  304.     {"0currentcolorrendering", zcurrentcolorrendering},
  305.     {"1setcolorrendering", zsetcolorrendering},
  306.         /* Internal operators */
  307.     {"1%cie_render_finish", cie_cache_render_finish},
  308.     {"3%cie_exec_tpqr", cie_exec_tpqr},
  309.     {"2%cie_post_exec_tpqr", cie_post_exec_tpqr},
  310.     {"1%cie_tpqr_finish", cie_tpqr_finish},
  311. END_OP_DEFS(0) }
  312.